home *** CD-ROM | disk | FTP | other *** search
- /* Internet User Data Protocol (UDP)
- * Copyright 1991 Phil Karn, KA9Q
- */
- #include "global.h"
- #include "mbuf.h"
- #include "netuser.h"
- #include "iface.h"
- #include "udp.h"
- #include "icmp.h"
-
- #if !defined(_lint)
- static char rcsid[] OPTIONAL = "$Id: udp.c,v 1.15 1997/09/07 00:31:16 root Exp root $";
- #endif
-
- static struct udp_cb *lookup_udp (struct socket *thesocket);
-
- struct mib_entry Udp_mib[] = {
- { "", { 0 } },
- { "udpInDatagrams", { 0 } },
- { "udpNoPorts", { 0 } },
- { "udpInErrors", { 0 } },
- { "udpOutDatagrams", { 0 } },
- };
-
-
- /* UDP control structures list */
- struct udp_cb *Udps;
-
- /* Create a UDP control block for lsocket, so that we can queue
- * incoming datagrams.
- */
- struct udp_cb *
- open_udp (lsocket, r_upcall)
- struct socket *lsocket;
- void (*r_upcall)(struct iface *,struct udp_cb *,int16);
- {
- register struct udp_cb *up;
-
- if ((up = lookup_udp (lsocket)) != NULLUDP) {
- /* Already exists */
- Net_error = CON_EXISTS;
- return NULLUDP;
- }
- up = (struct udp_cb *)callocw (1, sizeof (struct udp_cb));
- up->socket.address = lsocket->address;
- up->socket.port = lsocket->port;
- up->r_upcall = r_upcall;
-
- up->next = Udps;
- Udps = up;
- return up;
- }
-
-
-
- /* Send a UDP datagram */
- int
- send_udp (lsocket, fsocket, tos, ttl, data, length, id, df)
- struct socket *lsocket; /* Source socket */
- struct socket *fsocket; /* Destination socket */
- char tos; /* Type-of-service for IP */
- char ttl; /* Time-to-live for IP */
- struct mbuf *data; /* Data field, if any */
- int16 length; /* Length of data field */
- int16 id; /* Optional ID field for IP */
- char df; /* Don't Fragment flag for IP */
- {
- struct mbuf *bp;
- struct pseudo_header ph;
- struct udp udp;
- uint32 laddr;
-
- if (length != 0 && data != NULLBUF)
- trim_mbuf (&data, length);
- else
- length = len_p (data);
-
- length += UDPHDR;
-
- laddr = lsocket->address;
- if (laddr == INADDR_ANY)
- laddr = locaddr (fsocket->address);
-
- udp.source = lsocket->port;
- udp.dest = fsocket->port;
- udp.length = length;
-
- /* Create IP pseudo-header, compute checksum and send it */
- ph.length = length;
- ph.source = laddr;
- ph.dest = fsocket->address;
- ph.protocol = UDP_PTCL;
-
- if ((bp = htonudp (&udp, data, &ph)) == NULLBUF) {
- Net_error = NO_MEM;
- free_p (data);
- return 0;
- }
- udpOutDatagrams++;
- (void) ip_send (laddr, fsocket->address, UDP_PTCL, tos, ttl, bp, length, id, df);
- return (int)length;
- }
-
-
- /* Accept a waiting datagram, if available. Returns length of datagram */
- int
- recv_udp (up, fsocket, bp)
- register struct udp_cb *up;
- struct socket *fsocket; /* Place to stash incoming socket */
- struct mbuf **bp; /* Place to stash data packet */
- {
- struct socket sp;
- struct mbuf *buf;
- int16 length;
-
- if (up == NULLUDP) {
- Net_error = NO_CONN;
- return -1;
- }
- if (up->rcvcnt == 0) {
- Net_error = WOULDBLK;
- return -1;
- }
- buf = dequeue (&up->rcvq);
- up->rcvcnt--;
-
- /* Strip socket header */
- (void) pullup (&buf, (unsigned char *)&sp, sizeof(struct socket));
-
- /* Fill in the user's foreign socket structure, if given */
- if (fsocket != NULLSOCK) {
- fsocket->address = sp.address;
- fsocket->port = sp.port;
- }
- /* Hand data to user */
- length = len_p (buf);
- if (bp != NULLBUFP)
- *bp = buf;
- else
- free_p (buf);
- return (int)length;
- }
-
-
- /* Delete a UDP control block */
- int
- del_udp (conn)
- struct udp_cb *conn;
- {
- struct mbuf *bp;
- register struct udp_cb *up;
- struct udp_cb *udplast = NULLUDP;
-
- for (up = Udps; up != NULLUDP; udplast = up, up = up->next) {
- if (up == conn)
- break;
- }
- if (up == NULLUDP) {
- /* Either conn was NULL or not found on list */
- Net_error = INVALID;
- return -1;
- }
- /* Get rid of any pending packets */
- while (up->rcvcnt != 0) {
- bp = up->rcvq;
- up->rcvq = up->rcvq->anext;
- free_p (bp);
- up->rcvcnt--;
- }
- /* Remove from list */
- if (udplast != NULLUDP)
- udplast->next = up->next;
- else
- Udps = up->next; /* was first on list */
-
- free ((char *)up);
- return 0;
- }
-
-
- /* Process an incoming UDP datagram */
- void
- udp_input (iface, ip, bp, rxbroadcast)
- struct iface *iface; /* Input interface */
- struct ip *ip; /* IP header */
- struct mbuf *bp; /* UDP header and data */
- int rxbroadcast; /* The only protocol that accepts 'em */
- {
- struct pseudo_header ph;
- struct udp udp;
- struct udp_cb *up;
- struct socket lsocket;
- struct socket fsocket;
- struct mbuf *packet;
- int16 length;
-
- if (bp == NULLBUF)
- return;
-
- /* Create pseudo-header and verify checksum */
- ph.source = ip->source;
- ph.dest = ip->dest;
- ph.protocol = ip->protocol;
- length = ip->length - IPLEN - ip->optlen;
- ph.length = length;
-
- /* Peek at header checksum before we extract the header. This
- * allows us to bypass cksum() if the checksum field was not
- * set by the sender.
- */
- udp.checksum = udpcksum (bp);
- if (udp.checksum != 0 && cksum (&ph, bp, length) != 0) {
- /* Checksum non-zero, and wrong */
- udpInErrors++;
- free_p (bp);
- return;
- }
- /* Extract UDP header in host order */
- if (ntohudp (&udp, &bp) != 0) {
- /* Truncated header */
- udpInErrors++;
- free_p (bp);
- return;
- }
- /* If this was a broadcast packet, pretend it was sent to us */
- if (rxbroadcast)
- lsocket.address = iface->addr;
- else
- lsocket.address = ip->dest;
-
- lsocket.port = udp.dest;
- /* See if there's somebody around to read it */
- if ((up = lookup_udp (&lsocket)) == NULLUDP
- #ifdef UDPACCESS
- || tcp_check (UDPaccess, ip->source, udp.dest) > 0
- #endif
- ) {
- /* Nope, return an ICMP message */
- if (!rxbroadcast) {
- bp = htonudp (&udp, bp, &ph);
- (void) icmp_output (ip, bp, ICMP_DEST_UNREACH, ICMP_PORT_UNREACH, NULL);
- }
- udpNoPorts++;
- free_p (bp);
- return;
- }
- /* Create space for the foreign socket info */
- if ((packet = pushdown (bp, sizeof(fsocket))) == NULLBUF) {
- /* No space, drop whole packet */
- free_p (bp);
- udpInErrors++;
- return;
- }
- fsocket.address = ip->source;
- fsocket.port = udp.source;
- memcpy (&packet->data[0], (char *)&fsocket, sizeof(fsocket));
-
- /* Queue it */
- enqueue (&up->rcvq,packet);
- up->rcvcnt++;
- udpInDatagrams++;
- if (up->r_upcall)
- (*up->r_upcall) (iface, up, (int16) up->rcvcnt);
- }
-
-
- /* Look up UDP socket.
- * Return control block pointer or NULLUDP if nonexistant
- * As side effect, move control block to top of list to speed future
- * searches.
- */
- static struct udp_cb *
- lookup_udp (thesocket)
- struct socket *thesocket;
- {
- register struct udp_cb *up;
- struct udp_cb *uplast = NULLUDP;
-
- for (up = Udps; up != NULLUDP; uplast = up, up = up->next) {
- if (thesocket->port == up->socket.port && (thesocket->address == up->socket.address || up->socket.address == INADDR_ANY)) {
- if (uplast != NULLUDP) {
- /* Move to top of list */
- uplast->next = up->next;
- up->next = Udps;
- Udps = up;
- }
- return up;
- }
- }
- return NULLUDP;
- }
-
-
-
- #ifdef MSDOS
-
- /* Attempt to reclaim unused space in UDP receive queues */
- void
- udp_garbage (red)
- int red;
- {
- register struct udp_cb *udp;
-
- for (udp = Udps; udp != NULLUDP; udp = udp->next)
- mbuf_crunch (&udp->rcvq);
- }
-
- #endif
-